web3-eth
Version:
Web3 module to interact with the Ethereum blockchain and smart contracts.
1,125 lines (1,021 loc) • 37.3 kB
text/typescript
/*
This file is part of web3.js.
web3.js is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
web3.js is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with web3.js. If not, see <http://www.gnu.org/licenses/>.
*/
// Disabling because returnTypes must be last param to match 1.x params
/* eslint-disable default-param-last */
import {
ETH_DATA_FORMAT,
FormatType,
DataFormat,
DEFAULT_RETURN_FORMAT,
EthExecutionAPI,
SignedTransactionInfoAPI,
Web3BaseWalletAccount,
Address,
BlockTag,
BlockNumberOrTag,
Bytes,
Filter,
HexString,
Numbers,
HexStringBytes,
AccountObject,
Block,
FeeHistory,
Log,
TransactionReceipt,
Transaction,
TransactionCall,
Web3EthExecutionAPI,
TransactionWithFromLocalWalletIndex,
TransactionWithToLocalWalletIndex,
TransactionWithFromAndToLocalWalletIndex,
TransactionForAccessList,
AccessListResult,
Eip712TypedData,
} from 'web3-types';
import { Web3Context, Web3PromiEvent } from 'web3-core';
import { format, hexToBytes, bytesToUint8Array, numberToHex } from 'web3-utils';
import { TransactionFactory } from 'web3-eth-accounts';
import { isBlockTag, isBytes, isNullish, isString } from 'web3-validator';
import { SignatureError } from 'web3-errors';
import { ethRpcMethods } from 'web3-rpc-methods';
import { decodeSignedTransaction } from './utils/decode_signed_transaction.js';
import {
accountSchema,
blockSchema,
feeHistorySchema,
logSchema,
transactionReceiptSchema,
accessListResultSchema,
SignatureObjectSchema,
} from './schemas.js';
import {
SendSignedTransactionEvents,
SendSignedTransactionOptions,
SendTransactionEvents,
SendTransactionOptions,
TransactionMiddleware,
} from './types.js';
// eslint-disable-next-line import/no-cycle
import { getTransactionFromOrToAttr } from './utils/transaction_builder.js';
import { formatTransaction } from './utils/format_transaction.js';
// eslint-disable-next-line import/no-cycle
import { trySendTransaction } from './utils/try_send_transaction.js';
// eslint-disable-next-line import/no-cycle
import { waitForTransactionReceipt } from './utils/wait_for_transaction_receipt.js';
import { NUMBER_DATA_FORMAT } from './constants.js';
// eslint-disable-next-line import/no-cycle
import { SendTxHelper } from './utils/send_tx_helper.js';
/**
* View additional documentations here: {@link Web3Eth.getProtocolVersion}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export const getProtocolVersion = async (web3Context: Web3Context<EthExecutionAPI>) =>
ethRpcMethods.getProtocolVersion(web3Context.requestManager);
// TODO Add returnFormat parameter
/**
* View additional documentations here: {@link Web3Eth.isSyncing}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export const isSyncing = async (web3Context: Web3Context<EthExecutionAPI>) =>
ethRpcMethods.getSyncing(web3Context.requestManager);
// TODO consider adding returnFormat parameter (to format address as bytes)
/**
* View additional documentations here: {@link Web3Eth.getCoinbase}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export const getCoinbase = async (web3Context: Web3Context<EthExecutionAPI>) =>
ethRpcMethods.getCoinbase(web3Context.requestManager);
/**
* View additional documentations here: {@link Web3Eth.isMining}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export const isMining = async (web3Context: Web3Context<EthExecutionAPI>) =>
ethRpcMethods.getMining(web3Context.requestManager);
/**
* View additional documentations here: {@link Web3Eth.getHashRate}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getHashRate<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getHashRate(web3Context.requestManager);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getGasPrice}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getGasPrice<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getGasPrice(web3Context.requestManager);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getMaxPriorityFeePerGas}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getMaxPriorityFeePerGas<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getMaxPriorityFeePerGas(web3Context.requestManager);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getBlockNumber}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getBlockNumber<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getBlockNumber(web3Context.requestManager);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getBalance}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getBalance<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
address: Address,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.getBalance(
web3Context.requestManager,
address,
blockNumberFormatted,
);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getStorageAt}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getStorageAt<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
address: Address,
storageSlot: Numbers,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const storageSlotFormatted = format({ format: 'uint' }, storageSlot, ETH_DATA_FORMAT);
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.getStorageAt(
web3Context.requestManager,
address,
storageSlotFormatted,
blockNumberFormatted,
);
return format(
{ format: 'bytes' },
response as Bytes,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getCode}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getCode<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
address: Address,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.getCode(
web3Context.requestManager,
address,
blockNumberFormatted,
);
return format(
{ format: 'bytes' },
response as Bytes,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getBlock}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getBlock<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
block: Bytes | BlockNumberOrTag = web3Context.defaultBlock,
hydrated = false,
returnFormat: ReturnFormat,
) {
let response;
if (isBytes(block)) {
const blockHashFormatted = format({ format: 'bytes32' }, block, ETH_DATA_FORMAT);
response = await ethRpcMethods.getBlockByHash(
web3Context.requestManager,
blockHashFormatted as HexString,
hydrated,
);
} else {
const blockNumberFormatted = isBlockTag(block as string)
? (block as BlockTag)
: format({ format: 'uint' }, block as Numbers, ETH_DATA_FORMAT);
response = await ethRpcMethods.getBlockByNumber(
web3Context.requestManager,
blockNumberFormatted,
hydrated,
);
}
const res = format(
blockSchema,
response as unknown as Block,
returnFormat ?? web3Context.defaultReturnFormat,
);
if (!isNullish(res)) {
const result = {
...res,
transactions: res.transactions ?? [],
};
return result;
}
return res;
}
/**
* View additional documentations here: {@link Web3Eth.getBlockTransactionCount}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getBlockTransactionCount<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
block: Bytes | BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
let response;
if (isBytes(block)) {
const blockHashFormatted = format({ format: 'bytes32' }, block, ETH_DATA_FORMAT);
response = await ethRpcMethods.getBlockTransactionCountByHash(
web3Context.requestManager,
blockHashFormatted as HexString,
);
} else {
const blockNumberFormatted = isBlockTag(block as string)
? (block as BlockTag)
: format({ format: 'uint' }, block as Numbers, ETH_DATA_FORMAT);
response = await ethRpcMethods.getBlockTransactionCountByNumber(
web3Context.requestManager,
blockNumberFormatted,
);
}
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getBlockUncleCount}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getBlockUncleCount<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
block: Bytes | BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
let response;
if (isBytes(block)) {
const blockHashFormatted = format({ format: 'bytes32' }, block, ETH_DATA_FORMAT);
response = await ethRpcMethods.getUncleCountByBlockHash(
web3Context.requestManager,
blockHashFormatted as HexString,
);
} else {
const blockNumberFormatted = isBlockTag(block as string)
? (block as BlockTag)
: format({ format: 'uint' }, block as Numbers, ETH_DATA_FORMAT);
response = await ethRpcMethods.getUncleCountByBlockNumber(
web3Context.requestManager,
blockNumberFormatted,
);
}
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getUncle}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getUncle<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
block: Bytes | BlockNumberOrTag = web3Context.defaultBlock,
uncleIndex: Numbers,
returnFormat: ReturnFormat,
) {
const uncleIndexFormatted = format({ format: 'uint' }, uncleIndex, ETH_DATA_FORMAT);
let response;
if (isBytes(block)) {
const blockHashFormatted = format({ format: 'bytes32' }, block, ETH_DATA_FORMAT);
response = await ethRpcMethods.getUncleByBlockHashAndIndex(
web3Context.requestManager,
blockHashFormatted as HexString,
uncleIndexFormatted,
);
} else {
const blockNumberFormatted = isBlockTag(block as string)
? (block as BlockTag)
: format({ format: 'uint' }, block as Numbers, ETH_DATA_FORMAT);
response = await ethRpcMethods.getUncleByBlockNumberAndIndex(
web3Context.requestManager,
blockNumberFormatted,
uncleIndexFormatted,
);
}
return format(
blockSchema,
response as unknown as Block,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getTransaction}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getTransaction<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transactionHash: Bytes,
returnFormat: ReturnFormat = web3Context.defaultReturnFormat as ReturnFormat,
) {
const transactionHashFormatted = format(
{ format: 'bytes32' },
transactionHash,
DEFAULT_RETURN_FORMAT,
);
const response = await ethRpcMethods.getTransactionByHash(
web3Context.requestManager,
transactionHashFormatted,
);
return isNullish(response)
? response
: formatTransaction(response, returnFormat, {
transactionSchema: web3Context.config.customTransactionSchema,
fillInputAndData: true,
});
}
/**
* View additional documentations here: {@link Web3Eth.getPendingTransactions}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getPendingTransactions<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getPendingTransactions(web3Context.requestManager);
return response.map(transaction =>
formatTransaction(
transaction as unknown as Transaction,
returnFormat ?? web3Context.defaultReturnFormat,
{
transactionSchema: web3Context.config.customTransactionSchema,
fillInputAndData: true,
},
),
);
}
/**
* View additional documentations here: {@link Web3Eth.getTransactionFromBlock}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getTransactionFromBlock<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
block: Bytes | BlockNumberOrTag = web3Context.defaultBlock,
transactionIndex: Numbers,
returnFormat: ReturnFormat,
) {
const transactionIndexFormatted = format({ format: 'uint' }, transactionIndex, ETH_DATA_FORMAT);
let response;
if (isBytes(block)) {
const blockHashFormatted = format({ format: 'bytes32' }, block, ETH_DATA_FORMAT);
response = await ethRpcMethods.getTransactionByBlockHashAndIndex(
web3Context.requestManager,
blockHashFormatted as HexString,
transactionIndexFormatted,
);
} else {
const blockNumberFormatted = isBlockTag(block as string)
? (block as BlockTag)
: format({ format: 'uint' }, block as Numbers, ETH_DATA_FORMAT);
response = await ethRpcMethods.getTransactionByBlockNumberAndIndex(
web3Context.requestManager,
blockNumberFormatted,
transactionIndexFormatted,
);
}
return isNullish(response)
? response
: formatTransaction(response, returnFormat ?? web3Context.defaultReturnFormat, {
transactionSchema: web3Context.config.customTransactionSchema,
fillInputAndData: true,
});
}
/**
* View additional documentations here: {@link Web3Eth.getTransactionReceipt}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getTransactionReceipt<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transactionHash: Bytes,
returnFormat: ReturnFormat,
) {
const transactionHashFormatted = format(
{ format: 'bytes32' },
transactionHash,
DEFAULT_RETURN_FORMAT,
);
let response;
try {
response = await ethRpcMethods.getTransactionReceipt(
web3Context.requestManager,
transactionHashFormatted,
);
} catch (error) {
// geth indexing error, we poll until transactions stopped indexing
if (
typeof error === 'object' &&
!isNullish(error) &&
'message' in error &&
(error as { message: string }).message === 'transaction indexing is in progress'
) {
console.warn('Transaction indexing is in progress.');
} else {
throw error;
}
}
return isNullish(response)
? response
: format(
transactionReceiptSchema,
response as unknown as TransactionReceipt,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getTransactionCount}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getTransactionCount<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
address: Address,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.getTransactionCount(
web3Context.requestManager,
address,
blockNumberFormatted,
);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.sendTransaction}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export function sendTransaction<
ReturnFormat extends DataFormat,
ResolveType = FormatType<TransactionReceipt, ReturnFormat>,
>(
web3Context: Web3Context<EthExecutionAPI>,
transactionObj:
| Transaction
| TransactionWithFromLocalWalletIndex
| TransactionWithToLocalWalletIndex
| TransactionWithFromAndToLocalWalletIndex,
returnFormat: ReturnFormat,
options: SendTransactionOptions<ResolveType> = { checkRevertBeforeSending: true },
transactionMiddleware?: TransactionMiddleware,
): Web3PromiEvent<ResolveType, SendTransactionEvents<ReturnFormat>> {
const promiEvent = new Web3PromiEvent<ResolveType, SendTransactionEvents<ReturnFormat>>(
(resolve, reject) => {
setImmediate(() => {
(async () => {
const sendTxHelper = new SendTxHelper<ReturnFormat, ResolveType>({
web3Context,
promiEvent,
options,
returnFormat,
});
let transaction = { ...transactionObj };
if (!isNullish(transactionMiddleware)) {
transaction = await transactionMiddleware.processTransaction(transaction);
}
let transactionFormatted: FormatType<
| Transaction
| TransactionWithFromLocalWalletIndex
| TransactionWithToLocalWalletIndex
| TransactionWithFromAndToLocalWalletIndex,
ReturnFormat
> = formatTransaction(
{
...transaction,
from: getTransactionFromOrToAttr('from', web3Context, transaction),
to: getTransactionFromOrToAttr('to', web3Context, transaction),
},
ETH_DATA_FORMAT,
{
transactionSchema: web3Context.config.customTransactionSchema,
},
) as FormatType<Transaction, ReturnFormat>;
try {
transactionFormatted = (await sendTxHelper.populateGasPrice({
transaction,
transactionFormatted,
})) as FormatType<Transaction, ReturnFormat>;
await sendTxHelper.checkRevertBeforeSending(
transactionFormatted as TransactionCall,
);
sendTxHelper.emitSending(transactionFormatted);
let wallet: Web3BaseWalletAccount | undefined;
if (web3Context.wallet && !isNullish(transactionFormatted.from)) {
wallet = web3Context.wallet.get(
(transactionFormatted as Transaction).from as string,
);
}
const transactionHash: HexString = await sendTxHelper.signAndSend({
wallet,
tx: transactionFormatted,
});
const transactionHashFormatted = format(
{ format: 'bytes32' },
transactionHash as Bytes,
returnFormat ?? web3Context.defaultReturnFormat,
);
sendTxHelper.emitSent(transactionFormatted);
sendTxHelper.emitTransactionHash(
transactionHashFormatted as string & Uint8Array,
);
const transactionReceipt = await waitForTransactionReceipt(
web3Context,
transactionHash,
returnFormat ?? web3Context.defaultReturnFormat,
);
const transactionReceiptFormatted = sendTxHelper.getReceiptWithEvents(
format(
transactionReceiptSchema,
transactionReceipt,
returnFormat ?? web3Context.defaultReturnFormat,
),
);
sendTxHelper.emitReceipt(transactionReceiptFormatted);
resolve(
await sendTxHelper.handleResolve({
receipt: transactionReceiptFormatted,
tx: transactionFormatted as TransactionCall,
}),
);
sendTxHelper.emitConfirmation({
receipt: transactionReceiptFormatted,
transactionHash,
});
} catch (error) {
reject(
await sendTxHelper.handleError({
error,
tx: transactionFormatted as TransactionCall,
}),
);
}
})() as unknown;
});
},
);
return promiEvent;
}
/**
* View additional documentations here: {@link Web3Eth.sendSignedTransaction}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export function sendSignedTransaction<
ReturnFormat extends DataFormat,
ResolveType = FormatType<TransactionReceipt, ReturnFormat>,
>(
web3Context: Web3Context<EthExecutionAPI>,
signedTransaction: Bytes,
returnFormat: ReturnFormat,
options: SendSignedTransactionOptions<ResolveType> = { checkRevertBeforeSending: true },
): Web3PromiEvent<ResolveType, SendSignedTransactionEvents<ReturnFormat>> {
// TODO - Promise returned in function argument where a void return was expected
// eslint-disable-next-line @typescript-eslint/no-misused-promises
const promiEvent = new Web3PromiEvent<ResolveType, SendSignedTransactionEvents<ReturnFormat>>(
(resolve, reject) => {
setImmediate(() => {
(async () => {
const sendTxHelper = new SendTxHelper<ReturnFormat, ResolveType>({
web3Context,
promiEvent,
options,
returnFormat,
});
// Formatting signedTransaction to be send to RPC endpoint
const signedTransactionFormattedHex = format(
{ format: 'bytes' },
signedTransaction,
ETH_DATA_FORMAT,
);
const unSerializedTransaction = TransactionFactory.fromSerializedData(
bytesToUint8Array(hexToBytes(signedTransactionFormattedHex)),
);
const unSerializedTransactionWithFrom = {
...unSerializedTransaction.toJSON(),
// Some providers will default `from` to address(0) causing the error
// reported from `eth_call` to not be the reason the user's tx failed
// e.g. `eth_call` will return an Out of Gas error for a failed
// smart contract execution contract, because the sender, address(0),
// has no balance to pay for the gas of the transaction execution
from: unSerializedTransaction.getSenderAddress().toString(),
};
try {
const { v, r, s, ...txWithoutSigParams } = unSerializedTransactionWithFrom;
await sendTxHelper.checkRevertBeforeSending(
txWithoutSigParams as TransactionCall,
);
sendTxHelper.emitSending(signedTransactionFormattedHex);
const transactionHash = await trySendTransaction(
web3Context,
async (): Promise<string> =>
ethRpcMethods.sendRawTransaction(
web3Context.requestManager,
signedTransactionFormattedHex,
),
);
sendTxHelper.emitSent(signedTransactionFormattedHex);
const transactionHashFormatted = format(
{ format: 'bytes32' },
transactionHash as Bytes,
returnFormat ?? web3Context.defaultReturnFormat,
);
sendTxHelper.emitTransactionHash(
transactionHashFormatted as string & Uint8Array,
);
const transactionReceipt = await waitForTransactionReceipt(
web3Context,
transactionHash,
returnFormat ?? web3Context.defaultReturnFormat,
);
const transactionReceiptFormatted = sendTxHelper.getReceiptWithEvents(
format(
transactionReceiptSchema,
transactionReceipt,
returnFormat ?? web3Context.defaultReturnFormat,
),
);
sendTxHelper.emitReceipt(transactionReceiptFormatted);
resolve(
await sendTxHelper.handleResolve({
receipt: transactionReceiptFormatted,
tx: unSerializedTransactionWithFrom as TransactionCall,
}),
);
sendTxHelper.emitConfirmation({
receipt: transactionReceiptFormatted,
transactionHash,
});
} catch (error) {
reject(
await sendTxHelper.handleError({
error,
tx: unSerializedTransactionWithFrom as TransactionCall,
}),
);
}
})() as unknown;
});
},
);
return promiEvent;
}
/**
* View additional documentations here: {@link Web3Eth.sign}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function sign<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
message: Bytes,
addressOrIndex: Address | number,
returnFormat: ReturnFormat = web3Context.defaultReturnFormat as ReturnFormat,
) {
const messageFormatted = format({ format: 'bytes' }, message, DEFAULT_RETURN_FORMAT);
if (web3Context.wallet?.get(addressOrIndex)) {
const wallet = web3Context.wallet.get(addressOrIndex) as Web3BaseWalletAccount;
const signed = wallet.sign(messageFormatted);
return format(SignatureObjectSchema, signed, returnFormat);
}
if (typeof addressOrIndex === 'number') {
throw new SignatureError(
message,
'RPC method "eth_sign" does not support index signatures',
);
}
const response = await ethRpcMethods.sign(
web3Context.requestManager,
addressOrIndex,
messageFormatted,
);
return format({ format: 'bytes' }, response as Bytes, returnFormat);
}
/**
* View additional documentations here: {@link Web3Eth.signTransaction}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function signTransaction<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transaction: Transaction,
returnFormat: ReturnFormat = web3Context.defaultReturnFormat as ReturnFormat,
) {
const response = await ethRpcMethods.signTransaction(
web3Context.requestManager,
formatTransaction(transaction, ETH_DATA_FORMAT, {
transactionSchema: web3Context.config.customTransactionSchema,
}),
);
// Some clients only return the encoded signed transaction (e.g. Ganache)
// while clients such as Geth return the desired SignedTransactionInfoAPI object
return isString(response as HexStringBytes)
? decodeSignedTransaction(response as HexStringBytes, returnFormat, {
fillInputAndData: true,
})
: {
raw: format(
{ format: 'bytes' },
(response as SignedTransactionInfoAPI).raw,
returnFormat,
),
tx: formatTransaction((response as SignedTransactionInfoAPI).tx, returnFormat, {
transactionSchema: web3Context.config.customTransactionSchema,
fillInputAndData: true,
}),
};
}
// TODO Decide what to do with transaction.to
// https://github.com/ChainSafe/web3.js/pull/4525#issuecomment-982330076
/**
* View additional documentations here: {@link Web3Eth.call}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function call<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transaction: TransactionCall,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat = web3Context.defaultReturnFormat as ReturnFormat,
) {
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.call(
web3Context.requestManager,
formatTransaction(transaction, ETH_DATA_FORMAT, {
transactionSchema: web3Context.config.customTransactionSchema,
}),
blockNumberFormatted,
);
return format({ format: 'bytes' }, response as Bytes, returnFormat);
}
// TODO - Investigate whether response is padded as 1.x docs suggest
/**
* View additional documentations here: {@link Web3Eth.estimateGas}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function estimateGas<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transaction: Transaction,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const transactionFormatted = formatTransaction(transaction, ETH_DATA_FORMAT, {
transactionSchema: web3Context.config.customTransactionSchema,
});
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.estimateGas(
web3Context.requestManager,
transactionFormatted,
blockNumberFormatted,
);
return format(
{ format: 'uint' },
response as Numbers,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
// TODO - Add input formatting to filter
/**
* View additional documentations here: {@link Web3Eth.getPastLogs}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getLogs<ReturnFormat extends DataFormat>(
web3Context: Web3Context<Web3EthExecutionAPI>,
filter: Filter,
returnFormat: ReturnFormat,
) {
// format type bigint or number toBlock and fromBlock to hexstring.
let { toBlock, fromBlock } = filter;
if (!isNullish(toBlock)) {
if (typeof toBlock === 'number' || typeof toBlock === 'bigint') {
toBlock = numberToHex(toBlock);
}
}
if (!isNullish(fromBlock)) {
if (typeof fromBlock === 'number' || typeof fromBlock === 'bigint') {
fromBlock = numberToHex(fromBlock);
}
}
const formattedFilter = { ...filter, fromBlock, toBlock };
const response = await ethRpcMethods.getLogs(web3Context.requestManager, formattedFilter);
const result = response.map(res => {
if (typeof res === 'string') {
return res;
}
return format(
logSchema,
res as unknown as Log,
returnFormat ?? web3Context.defaultReturnFormat,
);
});
return result;
}
/**
* View additional documentations here: {@link Web3Eth.getChainId}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getChainId<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.getChainId(web3Context.requestManager);
return format(
{ format: 'uint' },
// Response is number in hex formatted string
response as unknown as number,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.getProof}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getProof<ReturnFormat extends DataFormat>(
web3Context: Web3Context<Web3EthExecutionAPI>,
address: Address,
storageKeys: Bytes[],
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const storageKeysFormatted = storageKeys.map(storageKey =>
format({ format: 'bytes' }, storageKey, ETH_DATA_FORMAT),
);
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = await ethRpcMethods.getProof(
web3Context.requestManager,
address,
storageKeysFormatted,
blockNumberFormatted,
);
return format(
accountSchema,
response as unknown as AccountObject,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
// TODO Throwing an error with Geth, but not Infura
// TODO gasUsedRatio and reward not formatting
/**
* View additional documentations here: {@link Web3Eth.getFeeHistory}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function getFeeHistory<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
blockCount: Numbers,
newestBlock: BlockNumberOrTag = web3Context.defaultBlock,
rewardPercentiles: Numbers[],
returnFormat: ReturnFormat,
) {
const blockCountFormatted = format({ format: 'uint' }, blockCount, ETH_DATA_FORMAT);
const newestBlockFormatted = isBlockTag(newestBlock as string)
? (newestBlock as BlockTag)
: format({ format: 'uint' }, newestBlock as Numbers, ETH_DATA_FORMAT);
const rewardPercentilesFormatted = format(
{
type: 'array',
items: {
format: 'uint',
},
},
rewardPercentiles,
NUMBER_DATA_FORMAT,
);
const response = await ethRpcMethods.getFeeHistory(
web3Context.requestManager,
blockCountFormatted,
newestBlockFormatted,
rewardPercentilesFormatted,
);
return format(
feeHistorySchema,
response as unknown as FeeHistory,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.createAccessList}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function createAccessList<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
transaction: TransactionForAccessList,
blockNumber: BlockNumberOrTag = web3Context.defaultBlock,
returnFormat: ReturnFormat,
) {
const blockNumberFormatted = isBlockTag(blockNumber as string)
? (blockNumber as BlockTag)
: format({ format: 'uint' }, blockNumber as Numbers, ETH_DATA_FORMAT);
const response = (await ethRpcMethods.createAccessList(
web3Context.requestManager,
formatTransaction(transaction, ETH_DATA_FORMAT, {
transactionSchema: web3Context.config.customTransactionSchema,
}),
blockNumberFormatted,
)) as unknown as AccessListResult;
return format(
accessListResultSchema,
response,
returnFormat ?? web3Context.defaultReturnFormat,
);
}
/**
* View additional documentations here: {@link Web3Eth.signTypedData}
* @param web3Context ({@link Web3Context}) Web3 configuration object that contains things such as the provider, request manager, wallet, etc.
*/
export async function signTypedData<ReturnFormat extends DataFormat>(
web3Context: Web3Context<EthExecutionAPI>,
address: Address,
typedData: Eip712TypedData,
useLegacy: boolean,
returnFormat: ReturnFormat,
) {
const response = await ethRpcMethods.signTypedData(
web3Context.requestManager,
address,
typedData,
useLegacy,
);
return format({ format: 'bytes' }, response, returnFormat ?? web3Context.defaultReturnFormat);
}