mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 13:17:39 +00:00
fix(providers): include provider name in billing error messages (#14697)
Merged via /review-pr -> /prepare-pr -> /merge-pr.
Prepared head SHA: 774e0b6605
Co-authored-by: fagemx <117356295+fagemx@users.noreply.github.com>
Co-authored-by: shakkernerd <165377636+shakkernerd@users.noreply.github.com>
Reviewed-by: @shakkernerd
This commit is contained in:
@@ -1,13 +1,36 @@
|
|||||||
import type { AssistantMessage } from "@mariozechner/pi-ai";
|
import type { AssistantMessage } from "@mariozechner/pi-ai";
|
||||||
import { describe, expect, it } from "vitest";
|
import { describe, expect, it } from "vitest";
|
||||||
import { BILLING_ERROR_USER_MESSAGE, formatAssistantErrorText } from "./pi-embedded-helpers.js";
|
import {
|
||||||
|
BILLING_ERROR_USER_MESSAGE,
|
||||||
|
formatBillingErrorMessage,
|
||||||
|
formatAssistantErrorText,
|
||||||
|
} from "./pi-embedded-helpers.js";
|
||||||
|
|
||||||
describe("formatAssistantErrorText", () => {
|
describe("formatAssistantErrorText", () => {
|
||||||
const makeAssistantError = (errorMessage: string): AssistantMessage =>
|
const makeAssistantError = (errorMessage: string): AssistantMessage => ({
|
||||||
({
|
role: "assistant",
|
||||||
stopReason: "error",
|
api: "openai-responses",
|
||||||
errorMessage,
|
provider: "openai",
|
||||||
}) as AssistantMessage;
|
model: "test-model",
|
||||||
|
usage: {
|
||||||
|
input: 0,
|
||||||
|
output: 0,
|
||||||
|
cacheRead: 0,
|
||||||
|
cacheWrite: 0,
|
||||||
|
totalTokens: 0,
|
||||||
|
cost: {
|
||||||
|
input: 0,
|
||||||
|
output: 0,
|
||||||
|
cacheRead: 0,
|
||||||
|
cacheWrite: 0,
|
||||||
|
total: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
stopReason: "error",
|
||||||
|
errorMessage,
|
||||||
|
content: [{ type: "text", text: errorMessage }],
|
||||||
|
timestamp: 0,
|
||||||
|
});
|
||||||
|
|
||||||
it("returns a friendly message for context overflow", () => {
|
it("returns a friendly message for context overflow", () => {
|
||||||
const msg = makeAssistantError("request_too_large");
|
const msg = makeAssistantError("request_too_large");
|
||||||
@@ -68,4 +91,17 @@ describe("formatAssistantErrorText", () => {
|
|||||||
const result = formatAssistantErrorText(msg);
|
const result = formatAssistantErrorText(msg);
|
||||||
expect(result).toBe(BILLING_ERROR_USER_MESSAGE);
|
expect(result).toBe(BILLING_ERROR_USER_MESSAGE);
|
||||||
});
|
});
|
||||||
|
it("includes provider name in billing message when provider is given", () => {
|
||||||
|
const msg = makeAssistantError("insufficient credits");
|
||||||
|
const result = formatAssistantErrorText(msg, { provider: "Anthropic" });
|
||||||
|
expect(result).toBe(formatBillingErrorMessage("Anthropic"));
|
||||||
|
expect(result).toContain("Anthropic");
|
||||||
|
expect(result).not.toContain("API provider");
|
||||||
|
});
|
||||||
|
it("returns generic billing message when provider is not given", () => {
|
||||||
|
const msg = makeAssistantError("insufficient credits");
|
||||||
|
const result = formatAssistantErrorText(msg);
|
||||||
|
expect(result).toContain("API provider");
|
||||||
|
expect(result).toBe(BILLING_ERROR_USER_MESSAGE);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ export {
|
|||||||
} from "./pi-embedded-helpers/bootstrap.js";
|
} from "./pi-embedded-helpers/bootstrap.js";
|
||||||
export {
|
export {
|
||||||
BILLING_ERROR_USER_MESSAGE,
|
BILLING_ERROR_USER_MESSAGE,
|
||||||
|
formatBillingErrorMessage,
|
||||||
classifyFailoverReason,
|
classifyFailoverReason,
|
||||||
formatRawAssistantErrorForUi,
|
formatRawAssistantErrorForUi,
|
||||||
formatAssistantErrorText,
|
formatAssistantErrorText,
|
||||||
|
|||||||
@@ -3,8 +3,15 @@ import type { OpenClawConfig } from "../../config/config.js";
|
|||||||
import type { FailoverReason } from "./types.js";
|
import type { FailoverReason } from "./types.js";
|
||||||
import { formatSandboxToolPolicyBlockedMessage } from "../sandbox.js";
|
import { formatSandboxToolPolicyBlockedMessage } from "../sandbox.js";
|
||||||
|
|
||||||
export const BILLING_ERROR_USER_MESSAGE =
|
export function formatBillingErrorMessage(provider?: string): string {
|
||||||
"⚠️ API provider returned a billing error — your API key has run out of credits or has an insufficient balance. Check your provider's billing dashboard and top up or switch to a different API key.";
|
const providerName = provider?.trim();
|
||||||
|
if (providerName) {
|
||||||
|
return `⚠️ ${providerName} returned a billing error — your API key has run out of credits or has an insufficient balance. Check your ${providerName} billing dashboard and top up or switch to a different API key.`;
|
||||||
|
}
|
||||||
|
return "⚠️ API provider returned a billing error — your API key has run out of credits or has an insufficient balance. Check your provider's billing dashboard and top up or switch to a different API key.";
|
||||||
|
}
|
||||||
|
|
||||||
|
export const BILLING_ERROR_USER_MESSAGE = formatBillingErrorMessage();
|
||||||
|
|
||||||
export function isContextOverflowError(errorMessage?: string): boolean {
|
export function isContextOverflowError(errorMessage?: string): boolean {
|
||||||
if (!errorMessage) {
|
if (!errorMessage) {
|
||||||
@@ -388,7 +395,7 @@ export function formatRawAssistantErrorForUi(raw?: string): string {
|
|||||||
|
|
||||||
export function formatAssistantErrorText(
|
export function formatAssistantErrorText(
|
||||||
msg: AssistantMessage,
|
msg: AssistantMessage,
|
||||||
opts?: { cfg?: OpenClawConfig; sessionKey?: string },
|
opts?: { cfg?: OpenClawConfig; sessionKey?: string; provider?: string },
|
||||||
): string | undefined {
|
): string | undefined {
|
||||||
// Also format errors if errorMessage is present, even if stopReason isn't "error"
|
// Also format errors if errorMessage is present, even if stopReason isn't "error"
|
||||||
const raw = (msg.errorMessage ?? "").trim();
|
const raw = (msg.errorMessage ?? "").trim();
|
||||||
@@ -450,7 +457,7 @@ export function formatAssistantErrorText(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isBillingErrorMessage(raw)) {
|
if (isBillingErrorMessage(raw)) {
|
||||||
return BILLING_ERROR_USER_MESSAGE;
|
return formatBillingErrorMessage(opts?.provider);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isLikelyHttpErrorText(raw) || isRawApiErrorPayload(raw)) {
|
if (isLikelyHttpErrorText(raw) || isRawApiErrorPayload(raw)) {
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ import {
|
|||||||
import { normalizeProviderId } from "../model-selection.js";
|
import { normalizeProviderId } from "../model-selection.js";
|
||||||
import { ensureOpenClawModelsJson } from "../models-config.js";
|
import { ensureOpenClawModelsJson } from "../models-config.js";
|
||||||
import {
|
import {
|
||||||
BILLING_ERROR_USER_MESSAGE,
|
formatBillingErrorMessage,
|
||||||
classifyFailoverReason,
|
classifyFailoverReason,
|
||||||
formatAssistantErrorText,
|
formatAssistantErrorText,
|
||||||
isAuthAssistantError,
|
isAuthAssistantError,
|
||||||
@@ -484,6 +484,7 @@ export async function runEmbeddedPiAgent(
|
|||||||
? formatAssistantErrorText(lastAssistant, {
|
? formatAssistantErrorText(lastAssistant, {
|
||||||
cfg: params.config,
|
cfg: params.config,
|
||||||
sessionKey: params.sessionKey ?? params.sessionId,
|
sessionKey: params.sessionKey ?? params.sessionId,
|
||||||
|
provider,
|
||||||
})
|
})
|
||||||
: undefined;
|
: undefined;
|
||||||
const assistantErrorText =
|
const assistantErrorText =
|
||||||
@@ -792,6 +793,7 @@ export async function runEmbeddedPiAgent(
|
|||||||
? formatAssistantErrorText(lastAssistant, {
|
? formatAssistantErrorText(lastAssistant, {
|
||||||
cfg: params.config,
|
cfg: params.config,
|
||||||
sessionKey: params.sessionKey ?? params.sessionId,
|
sessionKey: params.sessionKey ?? params.sessionId,
|
||||||
|
provider,
|
||||||
})
|
})
|
||||||
: undefined) ||
|
: undefined) ||
|
||||||
lastAssistant?.errorMessage?.trim() ||
|
lastAssistant?.errorMessage?.trim() ||
|
||||||
@@ -800,7 +802,7 @@ export async function runEmbeddedPiAgent(
|
|||||||
: rateLimitFailure
|
: rateLimitFailure
|
||||||
? "LLM request rate limited."
|
? "LLM request rate limited."
|
||||||
: billingFailure
|
: billingFailure
|
||||||
? BILLING_ERROR_USER_MESSAGE
|
? formatBillingErrorMessage(provider)
|
||||||
: authFailure
|
: authFailure
|
||||||
? "LLM request unauthorized."
|
? "LLM request unauthorized."
|
||||||
: "LLM request failed.");
|
: "LLM request failed.");
|
||||||
@@ -833,6 +835,7 @@ export async function runEmbeddedPiAgent(
|
|||||||
lastToolError: attempt.lastToolError,
|
lastToolError: attempt.lastToolError,
|
||||||
config: params.config,
|
config: params.config,
|
||||||
sessionKey: params.sessionKey ?? params.sessionId,
|
sessionKey: params.sessionKey ?? params.sessionId,
|
||||||
|
provider,
|
||||||
verboseLevel: params.verboseLevel,
|
verboseLevel: params.verboseLevel,
|
||||||
reasoningLevel: params.reasoningLevel,
|
reasoningLevel: params.reasoningLevel,
|
||||||
toolResultFormat: resolvedToolResultFormat,
|
toolResultFormat: resolvedToolResultFormat,
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import type { AssistantMessage } from "@mariozechner/pi-ai";
|
import type { AssistantMessage } from "@mariozechner/pi-ai";
|
||||||
import { describe, expect, it } from "vitest";
|
import { describe, expect, it } from "vitest";
|
||||||
|
import { formatBillingErrorMessage } from "../../pi-embedded-helpers.js";
|
||||||
import { buildEmbeddedRunPayloads } from "./payloads.js";
|
import { buildEmbeddedRunPayloads } from "./payloads.js";
|
||||||
|
|
||||||
describe("buildEmbeddedRunPayloads", () => {
|
describe("buildEmbeddedRunPayloads", () => {
|
||||||
@@ -14,13 +15,31 @@ describe("buildEmbeddedRunPayloads", () => {
|
|||||||
},
|
},
|
||||||
"request_id": "req_011CX7DwS7tSvggaNHmefwWg"
|
"request_id": "req_011CX7DwS7tSvggaNHmefwWg"
|
||||||
}`;
|
}`;
|
||||||
const makeAssistant = (overrides: Partial<AssistantMessage>): AssistantMessage =>
|
const makeAssistant = (overrides: Partial<AssistantMessage>): AssistantMessage => ({
|
||||||
({
|
role: "assistant",
|
||||||
stopReason: "error",
|
api: "openai-responses",
|
||||||
errorMessage: errorJson,
|
provider: "openai",
|
||||||
content: [{ type: "text", text: errorJson }],
|
model: "test-model",
|
||||||
...overrides,
|
usage: {
|
||||||
}) as AssistantMessage;
|
input: 0,
|
||||||
|
output: 0,
|
||||||
|
cacheRead: 0,
|
||||||
|
cacheWrite: 0,
|
||||||
|
totalTokens: 0,
|
||||||
|
cost: {
|
||||||
|
input: 0,
|
||||||
|
output: 0,
|
||||||
|
cacheRead: 0,
|
||||||
|
cacheWrite: 0,
|
||||||
|
total: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
timestamp: 0,
|
||||||
|
stopReason: "error",
|
||||||
|
errorMessage: errorJson,
|
||||||
|
content: [{ type: "text", text: errorJson }],
|
||||||
|
...overrides,
|
||||||
|
});
|
||||||
|
|
||||||
it("suppresses raw API error JSON when the assistant errored", () => {
|
it("suppresses raw API error JSON when the assistant errored", () => {
|
||||||
const lastAssistant = makeAssistant({});
|
const lastAssistant = makeAssistant({});
|
||||||
@@ -80,6 +99,27 @@ describe("buildEmbeddedRunPayloads", () => {
|
|||||||
expect(payloads.some((payload) => payload.text?.includes("request_id"))).toBe(false);
|
expect(payloads.some((payload) => payload.text?.includes("request_id"))).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("includes provider context for billing errors", () => {
|
||||||
|
const lastAssistant = makeAssistant({
|
||||||
|
errorMessage: "insufficient credits",
|
||||||
|
content: [{ type: "text", text: "insufficient credits" }],
|
||||||
|
});
|
||||||
|
const payloads = buildEmbeddedRunPayloads({
|
||||||
|
assistantTexts: [],
|
||||||
|
toolMetas: [],
|
||||||
|
lastAssistant,
|
||||||
|
sessionKey: "session:telegram",
|
||||||
|
provider: "Anthropic",
|
||||||
|
inlineToolResultsAllowed: false,
|
||||||
|
verboseLevel: "off",
|
||||||
|
reasoningLevel: "off",
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(payloads).toHaveLength(1);
|
||||||
|
expect(payloads[0]?.text).toBe(formatBillingErrorMessage("Anthropic"));
|
||||||
|
expect(payloads[0]?.isError).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
it("suppresses raw error JSON even when errorMessage is missing", () => {
|
it("suppresses raw error JSON even when errorMessage is missing", () => {
|
||||||
const lastAssistant = makeAssistant({ errorMessage: undefined });
|
const lastAssistant = makeAssistant({ errorMessage: undefined });
|
||||||
const payloads = buildEmbeddedRunPayloads({
|
const payloads = buildEmbeddedRunPayloads({
|
||||||
@@ -98,10 +138,15 @@ describe("buildEmbeddedRunPayloads", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("does not suppress error-shaped JSON when the assistant did not error", () => {
|
it("does not suppress error-shaped JSON when the assistant did not error", () => {
|
||||||
|
const lastAssistant = makeAssistant({
|
||||||
|
stopReason: "stop",
|
||||||
|
errorMessage: undefined,
|
||||||
|
content: [],
|
||||||
|
});
|
||||||
const payloads = buildEmbeddedRunPayloads({
|
const payloads = buildEmbeddedRunPayloads({
|
||||||
assistantTexts: [errorJsonPretty],
|
assistantTexts: [errorJsonPretty],
|
||||||
toolMetas: [],
|
toolMetas: [],
|
||||||
lastAssistant: { stopReason: "end_turn" } as AssistantMessage,
|
lastAssistant,
|
||||||
sessionKey: "session:telegram",
|
sessionKey: "session:telegram",
|
||||||
inlineToolResultsAllowed: false,
|
inlineToolResultsAllowed: false,
|
||||||
verboseLevel: "off",
|
verboseLevel: "off",
|
||||||
@@ -132,10 +177,15 @@ describe("buildEmbeddedRunPayloads", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("does not add tool error fallback when assistant output exists", () => {
|
it("does not add tool error fallback when assistant output exists", () => {
|
||||||
|
const lastAssistant = makeAssistant({
|
||||||
|
stopReason: "stop",
|
||||||
|
errorMessage: undefined,
|
||||||
|
content: [],
|
||||||
|
});
|
||||||
const payloads = buildEmbeddedRunPayloads({
|
const payloads = buildEmbeddedRunPayloads({
|
||||||
assistantTexts: ["All good"],
|
assistantTexts: ["All good"],
|
||||||
toolMetas: [],
|
toolMetas: [],
|
||||||
lastAssistant: { stopReason: "end_turn" } as AssistantMessage,
|
lastAssistant,
|
||||||
lastToolError: { toolName: "browser", error: "tab not found" },
|
lastToolError: { toolName: "browser", error: "tab not found" },
|
||||||
sessionKey: "session:telegram",
|
sessionKey: "session:telegram",
|
||||||
inlineToolResultsAllowed: false,
|
inlineToolResultsAllowed: false,
|
||||||
@@ -149,20 +199,22 @@ describe("buildEmbeddedRunPayloads", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("adds tool error fallback when the assistant only invoked tools", () => {
|
it("adds tool error fallback when the assistant only invoked tools", () => {
|
||||||
|
const lastAssistant = makeAssistant({
|
||||||
|
stopReason: "toolUse",
|
||||||
|
errorMessage: undefined,
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: "toolCall",
|
||||||
|
id: "toolu_01",
|
||||||
|
name: "exec",
|
||||||
|
arguments: { command: "echo hi" },
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
const payloads = buildEmbeddedRunPayloads({
|
const payloads = buildEmbeddedRunPayloads({
|
||||||
assistantTexts: [],
|
assistantTexts: [],
|
||||||
toolMetas: [],
|
toolMetas: [],
|
||||||
lastAssistant: {
|
lastAssistant,
|
||||||
stopReason: "toolUse",
|
|
||||||
content: [
|
|
||||||
{
|
|
||||||
type: "toolCall",
|
|
||||||
id: "toolu_01",
|
|
||||||
name: "exec",
|
|
||||||
arguments: { command: "echo hi" },
|
|
||||||
},
|
|
||||||
],
|
|
||||||
} as AssistantMessage,
|
|
||||||
lastToolError: { toolName: "exec", error: "Command exited with code 1" },
|
lastToolError: { toolName: "exec", error: "Command exited with code 1" },
|
||||||
sessionKey: "session:telegram",
|
sessionKey: "session:telegram",
|
||||||
inlineToolResultsAllowed: false,
|
inlineToolResultsAllowed: false,
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { parseReplyDirectives } from "../../../auto-reply/reply/reply-directives
|
|||||||
import { isSilentReplyText, SILENT_REPLY_TOKEN } from "../../../auto-reply/tokens.js";
|
import { isSilentReplyText, SILENT_REPLY_TOKEN } from "../../../auto-reply/tokens.js";
|
||||||
import { formatToolAggregate } from "../../../auto-reply/tool-meta.js";
|
import { formatToolAggregate } from "../../../auto-reply/tool-meta.js";
|
||||||
import {
|
import {
|
||||||
|
BILLING_ERROR_USER_MESSAGE,
|
||||||
formatAssistantErrorText,
|
formatAssistantErrorText,
|
||||||
formatRawAssistantErrorForUi,
|
formatRawAssistantErrorForUi,
|
||||||
getApiErrorPayloadFingerprint,
|
getApiErrorPayloadFingerprint,
|
||||||
@@ -27,6 +28,7 @@ export function buildEmbeddedRunPayloads(params: {
|
|||||||
lastToolError?: { toolName: string; meta?: string; error?: string };
|
lastToolError?: { toolName: string; meta?: string; error?: string };
|
||||||
config?: OpenClawConfig;
|
config?: OpenClawConfig;
|
||||||
sessionKey: string;
|
sessionKey: string;
|
||||||
|
provider?: string;
|
||||||
verboseLevel?: VerboseLevel;
|
verboseLevel?: VerboseLevel;
|
||||||
reasoningLevel?: ReasoningLevel;
|
reasoningLevel?: ReasoningLevel;
|
||||||
toolResultFormat?: ToolResultFormat;
|
toolResultFormat?: ToolResultFormat;
|
||||||
@@ -57,6 +59,7 @@ export function buildEmbeddedRunPayloads(params: {
|
|||||||
? formatAssistantErrorText(params.lastAssistant, {
|
? formatAssistantErrorText(params.lastAssistant, {
|
||||||
cfg: params.config,
|
cfg: params.config,
|
||||||
sessionKey: params.sessionKey,
|
sessionKey: params.sessionKey,
|
||||||
|
provider: params.provider,
|
||||||
})
|
})
|
||||||
: undefined;
|
: undefined;
|
||||||
const rawErrorMessage = lastAssistantErrored
|
const rawErrorMessage = lastAssistantErrored
|
||||||
@@ -75,6 +78,7 @@ export function buildEmbeddedRunPayloads(params: {
|
|||||||
? normalizeTextForComparison(rawErrorMessage)
|
? normalizeTextForComparison(rawErrorMessage)
|
||||||
: null;
|
: null;
|
||||||
const normalizedErrorText = errorText ? normalizeTextForComparison(errorText) : null;
|
const normalizedErrorText = errorText ? normalizeTextForComparison(errorText) : null;
|
||||||
|
const normalizedGenericBillingErrorText = normalizeTextForComparison(BILLING_ERROR_USER_MESSAGE);
|
||||||
const genericErrorText = "The AI service returned an error. Please try again.";
|
const genericErrorText = "The AI service returned an error. Please try again.";
|
||||||
if (errorText) {
|
if (errorText) {
|
||||||
replyItems.push({ text: errorText, isError: true });
|
replyItems.push({ text: errorText, isError: true });
|
||||||
@@ -133,6 +137,13 @@ export function buildEmbeddedRunPayloads(params: {
|
|||||||
if (trimmed === genericErrorText) {
|
if (trimmed === genericErrorText) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (
|
||||||
|
normalized &&
|
||||||
|
normalizedGenericBillingErrorText &&
|
||||||
|
normalized === normalizedGenericBillingErrorText
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (rawErrorMessage && trimmed === rawErrorMessage) {
|
if (rawErrorMessage && trimmed === rawErrorMessage) {
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
Reference in New Issue
Block a user