UNPKG

viem

Version:

TypeScript Interface for Ethereum

404 lines (377 loc) 12.8 kB
import type { Address } from 'abitype' import type { Account } from '../../accounts/types.js' import { type ParseAccountErrorType, parseAccount, } from '../../accounts/utils/parseAccount.js' import type { SignTransactionErrorType } from '../../accounts/utils/signTransaction.js' import type { Client } from '../../clients/createClient.js' import type { Transport } from '../../clients/transports/createTransport.js' import { AccountNotFoundError, type AccountNotFoundErrorType, AccountTypeNotSupportedError, type AccountTypeNotSupportedErrorType, } from '../../errors/account.js' import { BaseError } from '../../errors/base.js' import { TransactionReceiptRevertedError, type TransactionReceiptRevertedErrorType, } from '../../errors/transaction.js' import type { ErrorType } from '../../errors/utils.js' import type { GetAccountParameter } from '../../types/account.js' import type { Chain, DeriveChain, GetChainParameter, } from '../../types/chain.js' import type { GetTransactionRequestKzgParameter } from '../../types/kzg.js' import type { Hash } from '../../types/misc.js' import type { TransactionRequest } from '../../types/transaction.js' import type { UnionOmit } from '../../types/utils.js' import { type RecoverAuthorizationAddressErrorType, recoverAuthorizationAddress, } from '../../utils/authorization/recoverAuthorizationAddress.js' import type { RequestErrorType } from '../../utils/buildRequest.js' import { type AssertCurrentChainErrorType, assertCurrentChain, } from '../../utils/chain/assertCurrentChain.js' import { type GetTransactionErrorReturnType, getTransactionError, } from '../../utils/errors/getTransactionError.js' import { extract } from '../../utils/formatters/extract.js' import { type FormattedTransactionRequest, formatTransactionRequest, } from '../../utils/formatters/transactionRequest.js' import { getAction } from '../../utils/getAction.js' import { LruMap } from '../../utils/lru.js' import { type AssertRequestErrorType, type AssertRequestParameters, assertRequest, } from '../../utils/transaction/assertRequest.js' import { type GetChainIdErrorType, getChainId } from '../public/getChainId.js' import { type WaitForTransactionReceiptErrorType, waitForTransactionReceipt, } from '../public/waitForTransactionReceipt.js' import { defaultParameters, type PrepareTransactionRequestErrorType, prepareTransactionRequest, } from './prepareTransactionRequest.js' import { type SendRawTransactionSyncErrorType, type SendRawTransactionSyncReturnType, sendRawTransactionSync, } from './sendRawTransactionSync.js' const supportsWalletNamespace = new LruMap<boolean>(128) export type SendTransactionSyncRequest< chain extends Chain | undefined = Chain | undefined, chainOverride extends Chain | undefined = Chain | undefined, /// _derivedChain extends Chain | undefined = DeriveChain<chain, chainOverride>, > = UnionOmit<FormattedTransactionRequest<_derivedChain>, 'from'> & GetTransactionRequestKzgParameter export type SendTransactionSyncParameters< chain extends Chain | undefined = Chain | undefined, account extends Account | undefined = Account | undefined, chainOverride extends Chain | undefined = Chain | undefined, request extends SendTransactionSyncRequest< chain, chainOverride > = SendTransactionSyncRequest<chain, chainOverride>, > = request & GetAccountParameter<account, Account | Address, true, true> & GetChainParameter<chain, chainOverride> & GetTransactionRequestKzgParameter<request> & { /** Polling interval (ms) to poll for the transaction receipt. @default client.pollingInterval */ pollingInterval?: number | undefined /** Whether to throw an error if the transaction was detected as reverted. @default true */ throwOnReceiptRevert?: boolean | undefined /** Timeout (ms) to wait for a response. @default Math.max(chain.blockTime * 3, 5_000) */ timeout?: number | undefined } export type SendTransactionSyncReturnType< chain extends Chain | undefined = Chain | undefined, > = SendRawTransactionSyncReturnType<chain> export type SendTransactionSyncErrorType = | ParseAccountErrorType | GetTransactionErrorReturnType< | AccountNotFoundErrorType | AccountTypeNotSupportedErrorType | AssertCurrentChainErrorType | AssertRequestErrorType | GetChainIdErrorType | PrepareTransactionRequestErrorType | SendRawTransactionSyncErrorType | RecoverAuthorizationAddressErrorType | SignTransactionErrorType | TransactionReceiptRevertedErrorType | RequestErrorType > | WaitForTransactionReceiptErrorType | ErrorType /** * Creates, signs, and sends a new transaction to the network synchronously. * Returns the transaction receipt. * * @param client - Client to use * @param parameters - {@link SendTransactionSyncParameters} * @returns The transaction receipt. {@link SendTransactionSyncReturnType} * * @example * import { createWalletClient, custom } from 'viem' * import { mainnet } from 'viem/chains' * import { sendTransactionSync } from 'viem/wallet' * * const client = createWalletClient({ * chain: mainnet, * transport: custom(window.ethereum), * }) * const receipt = await sendTransactionSync(client, { * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', * value: 1000000000000000000n, * }) * * @example * // Account Hoisting * import { createWalletClient, http } from 'viem' * import { privateKeyToAccount } from 'viem/accounts' * import { mainnet } from 'viem/chains' * import { sendTransactionSync } from 'viem/wallet' * * const client = createWalletClient({ * account: privateKeyToAccount('0x…'), * chain: mainnet, * transport: http(), * }) * const receipt = await sendTransactionSync(client, { * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', * value: 1000000000000000000n, * }) */ export async function sendTransactionSync< chain extends Chain | undefined, account extends Account | undefined, const request extends SendTransactionSyncRequest<chain, chainOverride>, chainOverride extends Chain | undefined = undefined, >( client: Client<Transport, chain, account>, parameters: SendTransactionSyncParameters< chain, account, chainOverride, request >, ): Promise<SendTransactionSyncReturnType<chain>> { const { account: account_ = client.account, chain = client.chain, accessList, authorizationList, blobs, data, gas, gasPrice, maxFeePerBlobGas, maxFeePerGas, maxPriorityFeePerGas, nonce, pollingInterval, throwOnReceiptRevert, type, value, ...rest } = parameters const timeout = parameters.timeout ?? Math.max((chain?.blockTime ?? 0) * 3, 5_000) if (typeof account_ === 'undefined') throw new AccountNotFoundError({ docsPath: '/docs/actions/wallet/sendTransactionSync', }) const account = account_ ? parseAccount(account_) : null try { assertRequest(parameters as AssertRequestParameters) const to = await (async () => { // If `to` exists on the parameters, use that. if (parameters.to) return parameters.to // If `to` is null, we are sending a deployment transaction. if (parameters.to === null) return undefined // If no `to` exists, and we are sending a EIP-7702 transaction, use the // address of the first authorization in the list. if (authorizationList && authorizationList.length > 0) return await recoverAuthorizationAddress({ authorization: authorizationList[0], }).catch(() => { throw new BaseError( '`to` is required. Could not infer from `authorizationList`.', ) }) // Otherwise, we are sending a deployment transaction. return undefined })() if (account?.type === 'json-rpc' || account === null) { let chainId: number | undefined if (chain !== null) { chainId = await getAction(client, getChainId, 'getChainId')({}) assertCurrentChain({ currentChainId: chainId, chain, }) } const chainFormat = client.chain?.formatters?.transactionRequest?.format const format = chainFormat || formatTransactionRequest const request = format( { // Pick out extra data that might exist on the chain's transaction request type. ...extract(rest, { format: chainFormat }), accessList, account, authorizationList, blobs, chainId, data, gas, gasPrice, maxFeePerBlobGas, maxFeePerGas, maxPriorityFeePerGas, nonce, to, type, value, } as TransactionRequest, 'sendTransaction', ) const isWalletNamespaceSupported = supportsWalletNamespace.get(client.uid) const method = isWalletNamespaceSupported ? 'wallet_sendTransaction' : 'eth_sendTransaction' const hash = await (async () => { try { return await client.request( { method, params: [request], }, { retryCount: 0 }, ) } catch (e) { if (isWalletNamespaceSupported === false) throw e const error = e as BaseError // If the transport does not support the method or input, attempt to use the // `wallet_sendTransaction` method. if ( error.name === 'InvalidInputRpcError' || error.name === 'InvalidParamsRpcError' || error.name === 'MethodNotFoundRpcError' || error.name === 'MethodNotSupportedRpcError' ) { return (await client .request( { method: 'wallet_sendTransaction', params: [request], }, { retryCount: 0 }, ) .then((hash) => { supportsWalletNamespace.set(client.uid, true) return hash }) .catch((e) => { const walletNamespaceError = e as BaseError if ( walletNamespaceError.name === 'MethodNotFoundRpcError' || walletNamespaceError.name === 'MethodNotSupportedRpcError' ) { supportsWalletNamespace.set(client.uid, false) throw error } throw walletNamespaceError })) as never } throw error } })() const receipt = await getAction( client, waitForTransactionReceipt, 'waitForTransactionReceipt', )({ checkReplacement: false, hash, pollingInterval, timeout, }) if (throwOnReceiptRevert && receipt.status === 'reverted') throw new TransactionReceiptRevertedError({ receipt }) return receipt } if (account?.type === 'local') { // Prepare the request for signing (assign appropriate fees, etc.) const request = await getAction( client, prepareTransactionRequest, 'prepareTransactionRequest', )({ account, accessList, authorizationList, blobs, chain, data, gas, gasPrice, maxFeePerBlobGas, maxFeePerGas, maxPriorityFeePerGas, nonce, nonceManager: account.nonceManager, parameters: [...defaultParameters, 'sidecars'], type, value, ...rest, to, } as any) const serializer = chain?.serializers?.transaction const serializedTransaction = (await account.signTransaction(request, { serializer, })) as Hash return (await getAction( client, sendRawTransactionSync, 'sendRawTransactionSync', )({ serializedTransaction, throwOnReceiptRevert, timeout: parameters.timeout, })) as never } if (account?.type === 'smart') throw new AccountTypeNotSupportedError({ metaMessages: [ 'Consider using the `sendUserOperation` Action instead.', ], docsPath: '/docs/actions/bundler/sendUserOperation', type: 'smart', }) throw new AccountTypeNotSupportedError({ docsPath: '/docs/actions/wallet/sendTransactionSync', type: (account as any)?.type, }) } catch (err) { if (err instanceof AccountTypeNotSupportedError) throw err throw getTransactionError(err as BaseError, { ...parameters, account, chain: parameters.chain || undefined, }) } }