UNPKG

viem

Version:

TypeScript Interface for Ethereum

251 lines (244 loc) • 8.56 kB
import { BaseError } from '../errors/base.js' import { HttpRequestError, type HttpRequestErrorType, type RpcRequestErrorType, type TimeoutErrorType, type WebSocketRequestErrorType, } from '../errors/request.js' import { ChainDisconnectedError, type ChainDisconnectedErrorType, InternalRpcError, type InternalRpcErrorType, InvalidInputRpcError, type InvalidInputRpcErrorType, InvalidParamsRpcError, type InvalidParamsRpcErrorType, InvalidRequestRpcError, type InvalidRequestRpcErrorType, JsonRpcVersionUnsupportedError, type JsonRpcVersionUnsupportedErrorType, LimitExceededRpcError, type LimitExceededRpcErrorType, MethodNotFoundRpcError, type MethodNotFoundRpcErrorType, MethodNotSupportedRpcError, type MethodNotSupportedRpcErrorType, ParseRpcError, type ParseRpcErrorType, ProviderDisconnectedError, type ProviderDisconnectedErrorType, type ProviderRpcErrorCode, ResourceNotFoundRpcError, type ResourceNotFoundRpcErrorType, ResourceUnavailableRpcError, type ResourceUnavailableRpcErrorType, type RpcError, type RpcErrorCode, type RpcErrorType, SwitchChainError, type SwitchChainErrorType, TransactionRejectedRpcError, type TransactionRejectedRpcErrorType, UnauthorizedProviderError, type UnauthorizedProviderErrorType, UnknownRpcError, type UnknownRpcErrorType, UnsupportedProviderMethodError, type UnsupportedProviderMethodErrorType, UserRejectedRequestError, type UserRejectedRequestErrorType, } from '../errors/rpc.js' import type { ErrorType } from '../errors/utils.js' import type { EIP1193RequestFn, EIP1193RequestOptions, } from '../types/eip1193.js' import { stringToHex } from './encoding/toHex.js' import type { CreateBatchSchedulerErrorType } from './promise/createBatchScheduler.js' import { withDedupe } from './promise/withDedupe.js' import { type WithRetryErrorType, withRetry } from './promise/withRetry.js' import type { GetSocketRpcClientErrorType } from './rpc/socket.js' import { stringify } from './stringify.js' export type RequestErrorType = | ChainDisconnectedErrorType | CreateBatchSchedulerErrorType | HttpRequestErrorType | InternalRpcErrorType | InvalidInputRpcErrorType | InvalidParamsRpcErrorType | InvalidRequestRpcErrorType | GetSocketRpcClientErrorType | JsonRpcVersionUnsupportedErrorType | LimitExceededRpcErrorType | MethodNotFoundRpcErrorType | MethodNotSupportedRpcErrorType | ParseRpcErrorType | ProviderDisconnectedErrorType | ResourceNotFoundRpcErrorType | ResourceUnavailableRpcErrorType | RpcErrorType | RpcRequestErrorType | SwitchChainErrorType | TimeoutErrorType | TransactionRejectedRpcErrorType | UnauthorizedProviderErrorType | UnknownRpcErrorType | UnsupportedProviderMethodErrorType | UserRejectedRequestErrorType | WebSocketRequestErrorType | WithRetryErrorType | ErrorType export function buildRequest<request extends (args: any) => Promise<any>>( request: request, options: EIP1193RequestOptions = {}, ): EIP1193RequestFn { return async (args, overrideOptions = {}) => { const { dedupe = false, methods, retryDelay = 150, retryCount = 3, uid, } = { ...options, ...overrideOptions, } const { method } = args if (methods?.exclude?.includes(method)) throw new MethodNotSupportedRpcError(new Error('method not supported'), { method, }) if (methods?.include && !methods.include.includes(method)) throw new MethodNotSupportedRpcError(new Error('method not supported'), { method, }) const requestId = dedupe ? stringToHex(`${uid}.${stringify(args)}`) : undefined return withDedupe( () => withRetry( async () => { try { return await request(args) } catch (err_) { const err = err_ as unknown as RpcError< RpcErrorCode | ProviderRpcErrorCode > switch (err.code) { // -32700 case ParseRpcError.code: throw new ParseRpcError(err) // -32600 case InvalidRequestRpcError.code: throw new InvalidRequestRpcError(err) // -32601 case MethodNotFoundRpcError.code: throw new MethodNotFoundRpcError(err, { method: args.method }) // -32602 case InvalidParamsRpcError.code: throw new InvalidParamsRpcError(err) // -32603 case InternalRpcError.code: throw new InternalRpcError(err) // -32000 case InvalidInputRpcError.code: throw new InvalidInputRpcError(err) // -32001 case ResourceNotFoundRpcError.code: throw new ResourceNotFoundRpcError(err) // -32002 case ResourceUnavailableRpcError.code: throw new ResourceUnavailableRpcError(err) // -32003 case TransactionRejectedRpcError.code: throw new TransactionRejectedRpcError(err) // -32004 case MethodNotSupportedRpcError.code: throw new MethodNotSupportedRpcError(err, { method: args.method, }) // -32005 case LimitExceededRpcError.code: throw new LimitExceededRpcError(err) // -32006 case JsonRpcVersionUnsupportedError.code: throw new JsonRpcVersionUnsupportedError(err) // 4001 case UserRejectedRequestError.code: throw new UserRejectedRequestError(err) // 4100 case UnauthorizedProviderError.code: throw new UnauthorizedProviderError(err) // 4200 case UnsupportedProviderMethodError.code: throw new UnsupportedProviderMethodError(err) // 4900 case ProviderDisconnectedError.code: throw new ProviderDisconnectedError(err) // 4901 case ChainDisconnectedError.code: throw new ChainDisconnectedError(err) // 4902 case SwitchChainError.code: throw new SwitchChainError(err) // CAIP-25: User Rejected Error // https://docs.walletconnect.com/2.0/specs/clients/sign/error-codes#rejected-caip-25 case 5000: throw new UserRejectedRequestError(err) default: if (err_ instanceof BaseError) throw err_ throw new UnknownRpcError(err as Error) } } }, { delay: ({ count, error }) => { // If we find a Retry-After header, let's retry after the given time. if (error && error instanceof HttpRequestError) { const retryAfter = error?.headers?.get('Retry-After') if (retryAfter?.match(/\d/)) return Number.parseInt(retryAfter) * 1000 } // Otherwise, let's retry with an exponential backoff. return ~~(1 << count) * retryDelay }, retryCount, shouldRetry: ({ error }) => shouldRetry(error), }, ), { enabled: dedupe, id: requestId }, ) } } /** @internal */ export function shouldRetry(error: Error) { if ('code' in error && typeof error.code === 'number') { if (error.code === -1) return true // Unknown error if (error.code === LimitExceededRpcError.code) return true if (error.code === InternalRpcError.code) return true return false } if (error instanceof HttpRequestError && error.status) { // Forbidden if (error.status === 403) return true // Request Timeout if (error.status === 408) return true // Request Entity Too Large if (error.status === 413) return true // Too Many Requests if (error.status === 429) return true // Internal Server Error if (error.status === 500) return true // Bad Gateway if (error.status === 502) return true // Service Unavailable if (error.status === 503) return true // Gateway Timeout if (error.status === 504) return true return false } return true }