@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
text/typescript
/**
* 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
}