UNPKG

@tanstack/ai

Version:

Type-safe TypeScript AI SDK for streaming chat, tool calling, agents, structured outputs, and multimodal generation.

109 lines (105 loc) 4.28 kB
/** * Shared error-narrowing helper for activities that convert thrown values * into structured `RUN_ERROR` events. * * Accepts Error instances, objects with string-ish `message`/`code`, or bare * strings; always returns a shape safe to serialize. Never leaks the full * error object (which may carry request/response state from an SDK). * * Abort-shaped errors (DOM `AbortError`, OpenAI `APIUserAbortError`, * OpenRouter `RequestAbortedError`) are normalized to a stable * `{ message: 'Request aborted', code: 'aborted' }` shape so callers can * discriminate user-initiated cancellation from other failures without * matching on provider-specific message strings. */ const ABORT_ERROR_NAMES = new Set([ 'AbortError', 'APIUserAbortError', 'RequestAbortedError', ]) // HTTP status codes carried as numbers (e.g. `error.status = 429`) are a // common variant on SDK error classes; coerce so the resulting `code` field // is stable as a string for downstream consumers. function normalizeCode(codeField: unknown): string | undefined { if (typeof codeField === 'string') return codeField if (typeof codeField === 'number' && Number.isFinite(codeField)) { return String(codeField) } return undefined } export function toRunErrorPayload( error: unknown, fallbackMessage = 'Unknown error occurred', ): { message: string; code: string | undefined } { if (error && typeof error === 'object') { const name = (error as { name?: unknown }).name if (typeof name === 'string' && ABORT_ERROR_NAMES.has(name)) { return { message: 'Request aborted', code: 'aborted' } } } if (error instanceof Error) { const codeField = (error as Error & { code?: unknown }).code return { message: error.message || fallbackMessage, code: normalizeCode(codeField), } } if (typeof error === 'object' && error !== null) { const messageField = (error as { message?: unknown }).message const codeField = (error as { code?: unknown }).code return { message: typeof messageField === 'string' && messageField.length > 0 ? messageField : fallbackMessage, code: normalizeCode(codeField), } } if (typeof error === 'string' && error.length > 0) { return { message: error, code: undefined } } return { message: fallbackMessage, code: undefined } } /** * Extract the provider's *structured error body* from a thrown value, to attach * as the AG-UI `rawEvent` on a RUN_ERROR event. This is the recoverable upstream * detail (provider name, the upstream model's error JSON, rate-limit/overload * codes, etc.) that `toRunErrorPayload`'s `{ message, code }` deliberately drops. * * Security boundary: only known provider-response-body fields are forwarded — * never the raw SDK exception object, which can carry request metadata such as * auth headers or request ids. The recognized sources, in priority order: * * - `error.rawEvent` — a provider body an adapter attached explicitly (e.g. the * OpenRouter mid-stream `chunk.error`). * - `error.error` (object) — the parsed provider response body exposed by SDK * `APIError` instances (OpenAI/Anthropic `{ type, message, code, param }`, * OpenRouter typed errors whose `.error` carries `.metadata`). This is * provider-shaped data, distinct from `.headers` / `.request_id`. * - `error.metadata` — OpenRouter's `provider_name` + raw upstream body, when * surfaced directly on the thrown error. * * Returns `undefined` when no structured provider body is present, so callers * omit the field entirely rather than setting it to `null`: * * const rawEvent = toRunErrorRawEvent(error) * yield { type: EventType.RUN_ERROR, ..., ...(rawEvent !== undefined && { rawEvent }) } */ export function toRunErrorRawEvent(error: unknown): unknown { if (!error || typeof error !== 'object') return undefined const e = error as { rawEvent?: unknown error?: unknown metadata?: unknown } if (e.rawEvent !== undefined && e.rawEvent !== null) return e.rawEvent if ( e.error !== undefined && e.error !== null && typeof e.error === 'object' ) { return e.error } if (e.metadata !== undefined && e.metadata !== null) return e.metadata return undefined }