2026-01-14 16:52:10 -08:00
|
|
|
import type { AssistantMessage } from "@mariozechner/pi-ai";
|
|
|
|
|
import { describe, expect, it } from "vitest";
|
2026-02-13 02:23:27 +08:00
|
|
|
import { formatBillingErrorMessage } from "../../pi-embedded-helpers.js";
|
2026-02-19 08:53:50 +00:00
|
|
|
import { makeAssistantMessageFixture } from "../../test-helpers/assistant-message-fixtures.js";
|
2026-02-22 14:06:03 +00:00
|
|
|
import {
|
|
|
|
|
buildPayloads,
|
|
|
|
|
expectSinglePayloadText,
|
|
|
|
|
expectSingleToolErrorPayload,
|
|
|
|
|
} from "./payloads.test-helpers.js";
|
2026-01-14 16:52:10 -08:00
|
|
|
|
|
|
|
|
describe("buildEmbeddedRunPayloads", () => {
|
2026-02-22 14:06:03 +00:00
|
|
|
const OVERLOADED_FALLBACK_TEXT =
|
|
|
|
|
"The AI service is temporarily overloaded. Please try again in a moment.";
|
2026-01-15 01:34:19 +00:00
|
|
|
const errorJson =
|
2026-01-15 01:53:14 +00:00
|
|
|
'{"type":"error","error":{"details":null,"type":"overloaded_error","message":"Overloaded"},"request_id":"req_011CX7DwS7tSvggaNHmefwWg"}';
|
2026-01-15 01:34:19 +00:00
|
|
|
const errorJsonPretty = `{
|
|
|
|
|
"type": "error",
|
|
|
|
|
"error": {
|
|
|
|
|
"details": null,
|
|
|
|
|
"type": "overloaded_error",
|
|
|
|
|
"message": "Overloaded"
|
|
|
|
|
},
|
|
|
|
|
"request_id": "req_011CX7DwS7tSvggaNHmefwWg"
|
|
|
|
|
}`;
|
2026-02-19 08:53:50 +00:00
|
|
|
const makeAssistant = (overrides: Partial<AssistantMessage>): AssistantMessage =>
|
|
|
|
|
makeAssistantMessageFixture({
|
|
|
|
|
errorMessage: errorJson,
|
|
|
|
|
content: [{ type: "text", text: errorJson }],
|
|
|
|
|
...overrides,
|
|
|
|
|
});
|
2026-02-22 14:06:03 +00:00
|
|
|
const makeStoppedAssistant = () =>
|
|
|
|
|
makeAssistant({
|
|
|
|
|
stopReason: "stop",
|
|
|
|
|
errorMessage: undefined,
|
|
|
|
|
content: [],
|
2026-02-14 22:34:43 -05:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
const expectOverloadedFallback = (payloads: ReturnType<typeof buildPayloads>) => {
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.text).toBe(OVERLOADED_FALLBACK_TEXT);
|
|
|
|
|
};
|
|
|
|
|
|
2026-02-14 22:34:43 -05:00
|
|
|
it("suppresses raw API error JSON when the assistant errored", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
assistantTexts: [errorJson],
|
|
|
|
|
lastAssistant: makeAssistant({}),
|
2026-01-14 16:52:10 -08:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectOverloadedFallback(payloads);
|
2026-01-14 16:52:10 -08:00
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
expect(payloads.some((payload) => payload.text === errorJson)).toBe(false);
|
|
|
|
|
});
|
2026-01-15 01:34:19 +00:00
|
|
|
|
|
|
|
|
it("suppresses pretty-printed error JSON that differs from the errorMessage", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-15 01:34:19 +00:00
|
|
|
assistantTexts: [errorJsonPretty],
|
2026-02-14 22:34:43 -05:00
|
|
|
lastAssistant: makeAssistant({ errorMessage: errorJson }),
|
2026-01-15 01:34:19 +00:00
|
|
|
inlineToolResultsAllowed: true,
|
|
|
|
|
verboseLevel: "on",
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectOverloadedFallback(payloads);
|
2026-01-15 01:34:19 +00:00
|
|
|
expect(payloads.some((payload) => payload.text === errorJsonPretty)).toBe(false);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("suppresses raw error JSON from fallback assistant text", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
lastAssistant: makeAssistant({ content: [{ type: "text", text: errorJsonPretty }] }),
|
2026-01-15 01:34:19 +00:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectOverloadedFallback(payloads);
|
2026-01-15 01:34:19 +00:00
|
|
|
expect(payloads.some((payload) => payload.text?.includes("request_id"))).toBe(false);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-19 10:56:00 +08:00
|
|
|
it("includes provider and model context for billing errors", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
lastAssistant: makeAssistant({
|
2026-02-19 10:56:00 +08:00
|
|
|
model: "claude-3-5-sonnet",
|
2026-02-14 22:34:43 -05:00
|
|
|
errorMessage: "insufficient credits",
|
|
|
|
|
content: [{ type: "text", text: "insufficient credits" }],
|
|
|
|
|
}),
|
2026-02-13 02:23:27 +08:00
|
|
|
provider: "Anthropic",
|
2026-02-19 10:56:00 +08:00
|
|
|
model: "claude-3-5-sonnet",
|
2026-02-13 02:23:27 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
2026-02-19 10:56:00 +08:00
|
|
|
expect(payloads[0]?.text).toBe(formatBillingErrorMessage("Anthropic", "claude-3-5-sonnet"));
|
2026-02-13 02:23:27 +08:00
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-15 01:34:19 +00:00
|
|
|
it("suppresses raw error JSON even when errorMessage is missing", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-15 01:34:19 +00:00
|
|
|
assistantTexts: [errorJsonPretty],
|
2026-02-14 22:34:43 -05:00
|
|
|
lastAssistant: makeAssistant({ errorMessage: undefined }),
|
2026-01-15 01:34:19 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
expect(payloads.some((payload) => payload.text?.includes("request_id"))).toBe(false);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not suppress error-shaped JSON when the assistant did not error", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-15 01:34:19 +00:00
|
|
|
assistantTexts: [errorJsonPretty],
|
2026-02-22 14:06:03 +00:00
|
|
|
lastAssistant: makeStoppedAssistant(),
|
2026-01-15 01:34:19 +00:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectSinglePayloadText(payloads, errorJsonPretty.trim());
|
2026-01-15 01:34:19 +00:00
|
|
|
});
|
2026-01-18 18:35:03 +05:30
|
|
|
|
|
|
|
|
it("adds a fallback error when a tool fails and no assistant output exists", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-18 18:35:03 +05:30
|
|
|
lastToolError: { toolName: "browser", error: "tab not found" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
2026-01-23 00:59:44 +00:00
|
|
|
expect(payloads[0]?.text).toContain("Browser");
|
2026-01-18 18:35:03 +05:30
|
|
|
expect(payloads[0]?.text).toContain("tab not found");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not add tool error fallback when assistant output exists", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-18 18:35:03 +05:30
|
|
|
assistantTexts: ["All good"],
|
2026-02-22 14:06:03 +00:00
|
|
|
lastAssistant: makeStoppedAssistant(),
|
2026-01-18 18:35:03 +05:30
|
|
|
lastToolError: { toolName: "browser", error: "tab not found" },
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectSinglePayloadText(payloads, "All good");
|
2026-01-18 18:35:03 +05:30
|
|
|
});
|
2026-01-19 22:32:31 -08:00
|
|
|
|
2026-02-22 00:23:25 -08:00
|
|
|
it("adds completion fallback when tools run successfully without final assistant text", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
toolMetas: [{ toolName: "write", meta: "/tmp/out.md" }],
|
2026-02-22 14:06:03 +00:00
|
|
|
lastAssistant: makeStoppedAssistant(),
|
2026-02-22 00:23:25 -08:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectSinglePayloadText(payloads, "✅ Done.");
|
2026-02-22 00:23:25 -08:00
|
|
|
expect(payloads[0]?.isError).toBeUndefined();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not add completion fallback when the run still has a tool error", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
toolMetas: [{ toolName: "browser", meta: "open https://example.com" }],
|
|
|
|
|
lastToolError: { toolName: "browser", error: "url required" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not add completion fallback when no tools ran", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
2026-02-22 14:06:03 +00:00
|
|
|
lastAssistant: makeStoppedAssistant(),
|
2026-02-22 00:23:25 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-19 09:05:49 +05:30
|
|
|
it("adds tool error fallback when the assistant only invoked tools and verbose mode is on", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
lastAssistant: makeAssistant({
|
|
|
|
|
stopReason: "toolUse",
|
|
|
|
|
errorMessage: undefined,
|
|
|
|
|
content: [
|
|
|
|
|
{
|
|
|
|
|
type: "toolCall",
|
|
|
|
|
id: "toolu_01",
|
|
|
|
|
name: "exec",
|
|
|
|
|
arguments: { command: "echo hi" },
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
}),
|
2026-01-24 08:17:29 +00:00
|
|
|
lastToolError: { toolName: "exec", error: "Command exited with code 1" },
|
2026-02-19 09:05:49 +05:30
|
|
|
verboseLevel: "on",
|
2026-01-24 08:17:29 +00:00
|
|
|
});
|
|
|
|
|
|
2026-02-22 14:06:03 +00:00
|
|
|
expectSingleToolErrorPayload(payloads, {
|
|
|
|
|
title: "Exec",
|
|
|
|
|
detail: "code 1",
|
|
|
|
|
});
|
2026-01-24 08:17:29 +00:00
|
|
|
});
|
|
|
|
|
|
2026-02-16 21:24:34 +08:00
|
|
|
it("does not add tool error fallback when assistant text exists after tool calls", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
assistantTexts: ["Checked the page and recovered with final answer."],
|
|
|
|
|
lastAssistant: makeAssistant({
|
|
|
|
|
stopReason: "toolUse",
|
|
|
|
|
errorMessage: undefined,
|
|
|
|
|
content: [
|
|
|
|
|
{
|
|
|
|
|
type: "toolCall",
|
|
|
|
|
id: "toolu_01",
|
|
|
|
|
name: "browser",
|
|
|
|
|
arguments: { action: "search", query: "openclaw docs" },
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
}),
|
|
|
|
|
lastToolError: { toolName: "browser", error: "connection timeout" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBeUndefined();
|
|
|
|
|
expect(payloads[0]?.text).toContain("recovered");
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-14 23:01:16 +01:00
|
|
|
it("suppresses recoverable tool errors containing 'required' for non-mutating tools", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: { toolName: "browser", error: "url required" },
|
2026-01-19 22:32:31 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Recoverable errors should not be sent to the user
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-14 23:01:16 +01:00
|
|
|
it("suppresses recoverable tool errors containing 'missing' for non-mutating tools", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: { toolName: "browser", error: "url missing" },
|
2026-01-19 22:32:31 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-14 23:01:16 +01:00
|
|
|
it("suppresses recoverable tool errors containing 'invalid' for non-mutating tools", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: { toolName: "browser", error: "invalid parameter: url" },
|
2026-01-19 22:32:31 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-14 22:28:58 -05:00
|
|
|
it("suppresses non-mutating non-recoverable tool errors when messages.suppressToolErrors is enabled", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 22:28:58 -05:00
|
|
|
lastToolError: { toolName: "browser", error: "connection timeout" },
|
|
|
|
|
config: { messages: { suppressToolErrors: true } },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("still shows mutating tool errors when messages.suppressToolErrors is enabled", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 22:28:58 -05:00
|
|
|
lastToolError: { toolName: "write", error: "connection timeout" },
|
|
|
|
|
config: { messages: { suppressToolErrors: true } },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
expect(payloads[0]?.text).toContain("connection timeout");
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-16 13:29:24 -06:00
|
|
|
it("suppresses mutating tool errors when suppressToolErrorWarnings is enabled", () => {
|
|
|
|
|
const payloads = buildPayloads({
|
|
|
|
|
lastToolError: { toolName: "exec", error: "command not found" },
|
|
|
|
|
suppressToolErrorWarnings: true,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(0);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-14 23:01:16 +01:00
|
|
|
it("shows recoverable tool errors for mutating tools", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: { toolName: "message", meta: "reply", error: "text required" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
expect(payloads[0]?.text).toContain("required");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("shows mutating tool errors even when assistant output exists", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
assistantTexts: ["Done."],
|
2026-02-17 14:31:40 +09:00
|
|
|
lastAssistant: { stopReason: "end_turn" } as unknown as AssistantMessage,
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: { toolName: "write", error: "file missing" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(2);
|
|
|
|
|
expect(payloads[0]?.text).toBe("Done.");
|
|
|
|
|
expect(payloads[1]?.isError).toBe(true);
|
|
|
|
|
expect(payloads[1]?.text).toContain("missing");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not treat session_status read failures as mutating when explicitly flagged", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
assistantTexts: ["Status loaded."],
|
2026-02-17 14:31:40 +09:00
|
|
|
lastAssistant: { stopReason: "end_turn" } as unknown as AssistantMessage,
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: {
|
|
|
|
|
toolName: "session_status",
|
|
|
|
|
error: "model required",
|
|
|
|
|
mutatingAction: false,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.text).toBe("Status loaded.");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("dedupes identical tool warning text already present in assistant output", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const seed = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: {
|
|
|
|
|
toolName: "write",
|
|
|
|
|
error: "file missing",
|
|
|
|
|
mutatingAction: true,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const warningText = seed[0]?.text;
|
|
|
|
|
expect(warningText).toBeTruthy();
|
|
|
|
|
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-02-14 23:01:16 +01:00
|
|
|
assistantTexts: [warningText ?? ""],
|
2026-02-17 14:31:40 +09:00
|
|
|
lastAssistant: { stopReason: "end_turn" } as unknown as AssistantMessage,
|
2026-02-14 23:01:16 +01:00
|
|
|
lastToolError: {
|
|
|
|
|
toolName: "write",
|
|
|
|
|
error: "file missing",
|
|
|
|
|
mutatingAction: true,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.text).toBe(warningText);
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-19 22:32:31 -08:00
|
|
|
it("shows non-recoverable tool errors to the user", () => {
|
2026-02-14 22:34:43 -05:00
|
|
|
const payloads = buildPayloads({
|
2026-01-19 22:32:31 -08:00
|
|
|
lastToolError: { toolName: "browser", error: "connection timeout" },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Non-recoverable errors should still be shown
|
|
|
|
|
expect(payloads).toHaveLength(1);
|
|
|
|
|
expect(payloads[0]?.isError).toBe(true);
|
|
|
|
|
expect(payloads[0]?.text).toContain("connection timeout");
|
|
|
|
|
});
|
2026-01-14 16:52:10 -08:00
|
|
|
});
|