qwen-code/packages/core/src/utils/quotaErrorDetection.ts

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;
}