UNPKG

@lifi/composer-sdk

Version:

Public Composer SDK for building and submitting flows

151 lines (141 loc) 4.48 kB
import type { ComposeErrorKind, SimulationRevert } from '@lifi/compose-spec'; /** * Machine-readable error codes returned by the SDK. * * - `NETWORK_ERROR` — The HTTP request failed (DNS, timeout, connection refused). * - `VALIDATION_ERROR` — The server rejected the request (HTTP 400/422). * - `UNAUTHENTICATED` — The request lacks valid authentication credentials (HTTP 401). * - `FORBIDDEN` — The server understood the request but refuses to authorise it (HTTP 403). * - `SERVER_ERROR` — The server returned a 5xx status. * - `RATE_LIMITED` — The server returned HTTP 429. * - `NOT_FOUND` — The requested resource does not exist (HTTP 404). * - `UNKNOWN_ERROR` — An unexpected error that doesn't fit other categories. */ export type ComposeErrorCode = | 'NETWORK_ERROR' | 'VALIDATION_ERROR' | 'UNAUTHENTICATED' | 'FORBIDDEN' | 'SERVER_ERROR' | 'RATE_LIMITED' | 'NOT_FOUND' | 'UNKNOWN_ERROR'; /** * Error class for all failures originating from the Compose SDK or API. * * Includes structured metadata beyond the error message to support * programmatic error handling. * * @example * ```ts * try { * await builder.compile(run); * } catch (e) { * if (isComposeError(e) && e.code === 'VALIDATION_ERROR') { * console.error('Invalid request:', e.message, e.path); * } * } * ``` */ export class ComposeError extends Error { override readonly name = 'ComposeError'; /** Machine-readable error category. */ readonly code: ComposeErrorCode; /** HTTP status code, when the error originated from an HTTP response. */ readonly status?: number; /** The request URL that produced the error. */ readonly url?: string; /** Server-provided error kind for finer-grained classification. */ readonly kind?: ComposeErrorKind; /** JSON-pointer path to the field that caused a validation error. */ readonly path?: string; /** * Simulation revert diagnostics attached to `simulation_revert` errors. * Contains the raw error bytes and decoded error candidates when the * backend can parse the revert reason. */ readonly details?: SimulationRevert; constructor( code: ComposeErrorCode, message: string, options?: { status?: number; url?: string; cause?: unknown; kind?: ComposeErrorKind; path?: string; details?: SimulationRevert; }, ) { super(message, { cause: options?.cause }); this.code = code; this.status = options?.status; this.url = options?.url; this.kind = options?.kind; this.path = options?.path; this.details = options?.details; } } /** * Type guard that narrows an unknown error to {@link ComposeError}. * @param e - The value to check. * @returns `true` if `e` is an instance of `ComposeError`. */ export const isComposeError = (e: unknown): e is ComposeError => e instanceof ComposeError || (e instanceof Error && e.name === 'ComposeError' && 'code' in e); const STATUS_TO_CODE: ReadonlyMap<number, ComposeErrorCode> = new Map< number, ComposeErrorCode >([ [400, 'VALIDATION_ERROR'], [401, 'UNAUTHENTICATED'], [403, 'FORBIDDEN'], [404, 'NOT_FOUND'], [422, 'VALIDATION_ERROR'], [429, 'RATE_LIMITED'], ]); interface ServerErrorBody { readonly error?: { readonly kind?: string; readonly message?: string; readonly path?: string; readonly details?: SimulationRevert; }; } const tryParseErrorBody = (body: string): ServerErrorBody | null => { try { return JSON.parse(body) as ServerErrorBody; } catch { return null; } }; /** * Constructs a {@link ComposeError} from an HTTP error response, extracting * structured error details from the response body when available. * * @param status - The HTTP status code. * @param body - The raw response body text. * @param url - The request URL that produced the error. * @returns A `ComposeError` with the appropriate error code and metadata. */ export const errorFromHttpResponse = ( status: number, body: string, url: string, ): ComposeError => { const parsed = tryParseErrorBody(body); const serverError = parsed?.error; return new ComposeError( STATUS_TO_CODE.get(status) ?? (status >= 500 ? 'SERVER_ERROR' : 'UNKNOWN_ERROR'), (serverError?.message ?? body) || `HTTP ${status}`, { status, url, kind: serverError?.kind as ComposeErrorKind | undefined, path: serverError?.path, details: serverError?.details, }, ); };