167 lines
4.5 KiB
TypeScript
167 lines
4.5 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import type { StructuredError } from '../core/turn.js';
|
|
|
|
export interface ApiError {
|
|
error: {
|
|
code: number;
|
|
message: string;
|
|
status: string;
|
|
details: unknown[];
|
|
};
|
|
}
|
|
|
|
export function isApiError(error: unknown): error is ApiError {
|
|
return (
|
|
typeof error === 'object' &&
|
|
error !== null &&
|
|
'error' in error &&
|
|
typeof (error as ApiError).error === 'object' &&
|
|
'message' in (error as ApiError).error
|
|
);
|
|
}
|
|
|
|
export function isStructuredError(error: unknown): error is StructuredError {
|
|
return (
|
|
typeof error === 'object' &&
|
|
error !== null &&
|
|
'message' in error &&
|
|
typeof (error as StructuredError).message === 'string'
|
|
);
|
|
}
|
|
|
|
export function isProQuotaExceededError(error: unknown): boolean {
|
|
// Check for Pro quota exceeded errors by looking for the specific pattern
|
|
// This will match patterns like:
|
|
// - "Quota exceeded for quota metric 'Gemini 2.5 Pro Requests'"
|
|
// - "Quota exceeded for quota metric 'Gemini 2.5-preview Pro Requests'"
|
|
// We use string methods instead of regex to avoid ReDoS vulnerabilities
|
|
|
|
const checkMessage = (message: string): boolean =>
|
|
message.includes("Quota exceeded for quota metric 'Gemini") &&
|
|
message.includes("Pro Requests'");
|
|
|
|
if (typeof error === 'string') {
|
|
return checkMessage(error);
|
|
}
|
|
|
|
if (isStructuredError(error)) {
|
|
return checkMessage(error.message);
|
|
}
|
|
|
|
if (isApiError(error)) {
|
|
return checkMessage(error.error.message);
|
|
}
|
|
|
|
// Check if it's a Gaxios error with response data
|
|
if (error && typeof error === 'object' && 'response' in error) {
|
|
const gaxiosError = error as {
|
|
response?: {
|
|
data?: unknown;
|
|
};
|
|
};
|
|
if (gaxiosError.response && gaxiosError.response.data) {
|
|
if (typeof gaxiosError.response.data === 'string') {
|
|
return checkMessage(gaxiosError.response.data);
|
|
}
|
|
if (
|
|
typeof gaxiosError.response.data === 'object' &&
|
|
gaxiosError.response.data !== null &&
|
|
'error' in gaxiosError.response.data
|
|
) {
|
|
const errorData = gaxiosError.response.data as {
|
|
error?: { message?: string };
|
|
};
|
|
return checkMessage(errorData.error?.message || '');
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
export function isGenericQuotaExceededError(error: unknown): boolean {
|
|
if (typeof error === 'string') {
|
|
return error.includes('Quota exceeded for quota metric');
|
|
}
|
|
|
|
if (isStructuredError(error)) {
|
|
return error.message.includes('Quota exceeded for quota metric');
|
|
}
|
|
|
|
if (isApiError(error)) {
|
|
return error.error.message.includes('Quota exceeded for quota metric');
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export function isQwenQuotaExceededError(error: unknown): boolean {
|
|
// Check for Qwen insufficient quota errors (should not retry)
|
|
const checkMessage = (message: string): boolean => {
|
|
const lowerMessage = message.toLowerCase();
|
|
return (
|
|
lowerMessage.includes('insufficient_quota') ||
|
|
lowerMessage.includes('free allocated quota exceeded') ||
|
|
(lowerMessage.includes('quota') && lowerMessage.includes('exceeded'))
|
|
);
|
|
};
|
|
|
|
if (typeof error === 'string') {
|
|
return checkMessage(error);
|
|
}
|
|
|
|
if (isStructuredError(error)) {
|
|
return checkMessage(error.message);
|
|
}
|
|
|
|
if (isApiError(error)) {
|
|
return checkMessage(error.error.message);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export function isQwenThrottlingError(error: unknown): boolean {
|
|
// Check for Qwen throttling errors (should retry)
|
|
const checkMessage = (message: string): boolean => {
|
|
const lowerMessage = message.toLowerCase();
|
|
return (
|
|
lowerMessage.includes('throttling') ||
|
|
lowerMessage.includes('requests throttling triggered') ||
|
|
lowerMessage.includes('rate limit') ||
|
|
lowerMessage.includes('too many requests')
|
|
);
|
|
};
|
|
|
|
// Check status code
|
|
const getStatusCode = (error: unknown): number | undefined => {
|
|
if (error && typeof error === 'object') {
|
|
const errorObj = error as { status?: number; code?: number };
|
|
return errorObj.status || errorObj.code;
|
|
}
|
|
return undefined;
|
|
};
|
|
|
|
const statusCode = getStatusCode(error);
|
|
|
|
if (typeof error === 'string') {
|
|
return (
|
|
(statusCode === 429 && checkMessage(error)) ||
|
|
error.includes('throttling')
|
|
);
|
|
}
|
|
|
|
if (isStructuredError(error)) {
|
|
return statusCode === 429 && checkMessage(error.message);
|
|
}
|
|
|
|
if (isApiError(error)) {
|
|
return error.error.code === 429 && checkMessage(error.error.message);
|
|
}
|
|
|
|
return false;
|
|
}
|